<!--
// Copyright © 2020, 2021 Anticrm Platform Contributors.
// Copyright © 2021 Hardcore Engineering Inc.
//
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License. You may
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//
// See the License for the specific language governing permissions and
// limitations under the License.
-->
<script lang="ts">
  import type { Attachment } from '@hcengineering/attachment'
  import core, { type WithLookup } from '@hcengineering/core'
  import presentation, {
    FilePreviewPopup,
    canPreviewFile,
    getBlobRef,
    getFileUrl,
    getPreviewAlignment,
    previewTypes,
    sizeToWidth
  } from '@hcengineering/presentation'
  import { Label, closeTooltip, showPopup } from '@hcengineering/ui'
  import { permissionsStore } from '@hcengineering/view-resources'
  import filesize from 'filesize'
  import { createEventDispatcher } from 'svelte'
  import { getType } from '../utils'

  import AttachmentName from './AttachmentName.svelte'

  export let value: WithLookup<Attachment> | undefined
  export let removable: boolean = false
  export let showPreview = false
  export let preview = false
  export let progress: boolean = false

  const dispatch = createEventDispatcher()

  const maxLenght: number = 30
  const trimFilename = (fname: string): string =>
    fname.length > maxLenght ? fname.substr(0, (maxLenght - 1) / 2) + '...' + fname.substr(-(maxLenght - 1) / 2) : fname

  $: canRemove =
    removable &&
    value !== undefined &&
    value.readonly !== true &&
    ($permissionsStore.whitelist.has(value.space) ||
      !$permissionsStore.ps[value.space]?.has(core.permission.ForbidDeleteObject))

  function iconLabel (name: string): string {
    const parts = `${name}`.split('.')
    const ext = parts[parts.length - 1]
    return ext.substring(0, 4).toUpperCase()
  }
  function isImage (contentType: string): boolean {
    return getType(contentType) === 'image'
  }

  let canPreview: boolean = false
  $: if (value !== undefined) {
    void canPreviewFile(value.type, $previewTypes).then((res) => {
      canPreview = res
    })
  } else {
    canPreview = false
  }

  function clickHandler (e: MouseEvent): void {
    if (value === undefined || !canPreview) return

    e.preventDefault()
    e.stopPropagation()
    if (e.metaKey || e.ctrlKey) {
      window.open((e.target as HTMLAnchorElement).href, '_blank')
      return
    }
    closeTooltip()
    showPopup(
      FilePreviewPopup,
      {
        file: value.file,
        contentType: value.type,
        name: value.name,
        metadata: value.metadata
      },
      getPreviewAlignment(value.type)
    )
  }

  function middleClickHandler (e: MouseEvent): void {
    if (e.button !== 1) return
    e.preventDefault()
    e.stopPropagation()
    window.open((e.target as HTMLAnchorElement).href, '_blank')
  }

  let download: HTMLAnchorElement

  function dragStart (event: DragEvent): void {
    if (value === undefined) return
    const url = encodeURI(getFileUrl(value.file))
    event.dataTransfer?.setData('application/contentType', value.type)
    event.dataTransfer?.setData('text/plain', getFileUrl(value.file))
    event.dataTransfer?.setData('text/uri-list', url + '\r\n')
  }
</script>

{#if preview}
  <AttachmentName {value} />
{:else}
  <div class="flex-row-center attachment-container">
    {#if value}
      {#await getBlobRef(value.file, value.name, sizeToWidth('large')) then valueRef}
        <a
          class="no-line"
          style:flex-shrink={0}
          href={valueRef.src}
          download={value.name}
          on:click={clickHandler}
          on:mousedown={middleClickHandler}
          on:dragstart={dragStart}
        >
          {#if showPreview && isImage(value.type)}
            <img
              src={valueRef.src}
              data-id={value.file}
              srcset={valueRef.srcset}
              class="flex-center icon"
              class:svg={value.type === 'image/svg+xml'}
              class:image={isImage(value.type)}
              alt={value.name}
            />
          {:else}
            <div class="flex-center icon">
              {iconLabel(value.name)}
            </div>
          {/if}
        </a>
        <div class="flex-col info-container">
          <div class="name">
            <a href={valueRef.src} download={value.name} on:click={clickHandler} on:mousedown={middleClickHandler}>
              {trimFilename(value.name)}
            </a>
          </div>
          <div class="info-content flex-row-center">
            {filesize(value.size, { spacer: '' })}
            <span class="actions inline-flex clear-mins ml-1 gap-1">
              <span>•</span>
              <a class="no-line colorInherit" href={valueRef.src} download={value.name} bind:this={download}>
                <Label label={presentation.string.Download} />
              </a>
              {#if canRemove}
                <span>•</span>
                <!-- svelte-ignore a11y-click-events-have-key-events -->
                <!-- svelte-ignore a11y-no-static-element-interactions -->
                <span
                  class="remove-link"
                  on:click={(ev) => {
                    ev.stopPropagation()
                    ev.preventDefault()
                    dispatch('remove', value)
                  }}
                >
                  <Label label={presentation.string.Delete} />
                </span>
              {/if}
            </span>
          </div>
        </div>
      {/await}
    {/if}
  </div>
{/if}

<style lang="scss">
  .attachment-container {
    flex-shrink: 0;
    width: auto;
    height: 3rem;
    min-width: 17.25rem;
    border-radius: 0.25rem;

    .icon {
      flex-shrink: 0;
      width: 3rem;
      height: 3rem;
      object-fit: contain;
      border: 1px solid var(--theme-button-border);
      border-radius: 0.25rem 0 0 0.25rem;
      cursor: pointer;

      &:not(.image) {
        color: var(--primary-button-color);
        background-color: var(--primary-button-default);
      }
      &.svg {
        background-color: #fff;
      }
      &.image {
        background-position: center;
        background-repeat: no-repeat;
        background-size: cover;
      }
    }
    .info-container {
      padding: 0.5rem 0.75rem;
      width: 100%;
      height: 100%;
      background-color: var(--theme-button-default);
      border: 1px solid var(--theme-button-border);
      border-left: none;
      border-radius: 0 0.25rem 0.25rem 0;
    }
    .no-line:hover ~ .info-container,
    .info-container:hover {
      background-color: var(--theme-button-hovered);

      .actions {
        opacity: 1;
      }
    }
    .name {
      white-space: nowrap;
      font-size: 0.8125rem;
      color: var(--theme-caption-color);
      cursor: pointer;

      &:hover ~ .info-content .actions {
        opacity: 1;
      }
    }
    .info-content {
      white-space: nowrap;
      font-size: 0.6875rem;
      color: var(--theme-darker-color);

      .actions {
        opacity: 0;
        transition: opacity 0.1s var(--timing-main);
      }
      &:hover .actions {
        opacity: 1;
      }
    }
    .remove-link {
      color: var(--theme-error-color);
      cursor: pointer;

      &:hover {
        text-decoration-line: underline;
      }
    }
    a.colorInherit {
      color: inherit;

      &:hover {
        text-decoration: underline;
        color: var(--theme-dark-color);
      }
    }

    .name:hover,
    .no-line:hover + .info-container > .name a {
      text-decoration: underline;
      color: var(--theme-caption-color);
    }
    .name:active,
    .no-line:active + .info-container > .name a {
      text-decoration: underline;
      color: var(--theme-caption-color);
    }
  }
</style>
